MAIN PROJECT

ROUTER 1x3 IN SYSTEM VERILOG

SUBMITTED BY

MELVIN RIJOHN T

04/02/2025

**Table of Contents**

[**TESTBENCH FILES 3**](#)

[TRANSACTION 3](#_g6s72h4n8as1)

[GENERATOR 4](#_l3o358d3pzyq)

[DRIVER 6](#_4ewurvbou8y5)

[INTERFACE 8](#_uccmqk6ycb8o)

[MONITOR 8](#_k293jfntbqkr)

[SCOREBOARD 11](#_hdyrv6z0rwml)

[ENVIRONMENT 14](#_lmb1blcxjyew)

[TESTBENCH TOP 15](#_p3dqihqchaii)

[**DESIGN (ROUTER 1x3) 17**](#_ijudb335mec)

[ROUTER TOP 17](#_hkqrylll4293)

[FSM\_CONTROLLER 19](#_nqct2wvyfhlj)

[REGISTER 22](#_h9e5014nqqs6)

[FIFO 25](#_flum51x0yfbh)

[SYNCHRONIZER 28](#_7c7idild3kxu)

[**OUTPUT 31**](#_nrblf7hcu8um)

# **TESTBENCH FILES**

## TRANSACTION

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: transaction.sv                         \*/

/\*      DESCRIPTION: Formatting data packet               \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

// Packet type enumeration for defining packet states

typedef enum logic[1:0]{RESET = 0, HEADER = 1, PAYLOAD = 2, PARITY = 3} pkt\_type\_t;

// Packet class definition

class Packet;

    // Randomizable fields representing packet attributes

    rand bit[7:0] header;

    rand bit[7:0] data;

    rand bit pkt\_valid;

    randc bit rd\_en\_0;

    randc bit rd\_en\_1;

    randc bit rd\_en\_2;

    // Status signals

    bit vld\_out\_0;

    bit vld\_out\_1;

    bit vld\_out\_2;

    bit err;

    bit busy;

    bit rst;

    // Data output signals

    bit [7:0] dout\_0;

    bit [7:0] dout\_1;

    bit [7:0] dout\_2;

    // Parity check logic

    logic[7:0] parity;

    // Packet type enumeration variable

    pkt\_type\_t pkt\_type;

    // Constraints to ensure valid header values

    constraint con1 {

        header[1:0] != 2'b11;

        header[7:2] != 0;

        header[7:2] <= 20;

    }

    // Function to copy data from another Packet instance

    function void copy(Packet tmp);

        data = tmp.data;

        pkt\_valid = tmp.pkt\_valid;

        rd\_en\_0 = tmp.rd\_en\_0;

        rd\_en\_1 = tmp.rd\_en\_1;

        rd\_en\_2 = tmp.rd\_en\_2;

        vld\_out\_0 = tmp.vld\_out\_0;

        vld\_out\_1 = tmp.vld\_out\_1;

        vld\_out\_2 = tmp.vld\_out\_2;

        err = tmp.err;

        busy = tmp.busy;

        dout\_0 = tmp.dout\_0;

        dout\_1 = tmp.dout\_1;

        dout\_2 = tmp.dout\_2;

        parity = tmp.parity;

    endfunction

endclass

## GENERATOR

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: generator.sv                           \*/

/\*      DESCRIPTION: Generates input streams             \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

class Generator;

    mailbox #(Packet) mbx; // Mailbox for communication with driver

    event drv\_done; // Event to synchronize with driver

    Packet pkt; // Packet instance

    bit[7:0] header; // Header field storage

    function new(mailbox #(Packet) mbx, event drv\_done);

        this.mbx = mbx; // Initialize mailbox

        this.drv\_done = drv\_done; // Initialize event

    endfunction

    task run(int loopCount = 1);

        repeat(loopCount) begin

            pkt = new(); // Create a new packet instance

            pkt.parity = 0; // Initialize parity bit

            $display("[%0tps] Generator: Starting....", $time);

            pkt.pkt\_type = RESET; // Send reset packet

            mbx.put(pkt);

            @(drv\_done); // Wait for driver to complete

            if(!pkt.randomize()) $error("Randomization failed"); // Randomize packet fields

            pkt.pkt\_type = HEADER; // Set packet type to HEADER

            header = pkt.header; // Store header value

            pkt.parity = pkt.parity ^ pkt.header; // Compute parity

            mbx.put(pkt); // Send header packet

            @(drv\_done);

            for (int i = 0; i < header[7:2]; i++) begin // Loop through payload data

                if(!pkt.randomize()) $error("Randomization failed"); // Randomize payload data

                pkt.parity = pkt.parity ^ pkt.data; // Update parity

                pkt.pkt\_type = PAYLOAD; // Set packet type to PAYLOAD

                mbx.put(pkt); // Send payload packet

                @(drv\_done);

            end

            pkt.pkt\_type = PARITY; // Set packet type to PARITY

            pkt.data = pkt.parity; // Store computed parity in data field

            mbx.put(pkt); // Send parity packet

        end

    endtask

endclass

## DRIVER

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: driver.sv                             \*/

/\*      DESCRIPTION: Drives the input streams to the dut \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

class Driver;

    local bit[7:0] header; // Stores packet header data

    mailbox #(Packet) mbx; // Mailbox for communication with other components

    event drv\_done; // Event to signal when driving is done

    virtual router\_if vif; // Virtual interface for DUT interaction

    function new(mailbox #(Packet) mbx, event drv\_done, virtual router\_if vif);

        this.mbx = mbx; // Initialize mailbox

        this.drv\_done = drv\_done; // Initialize event

        this.vif = vif; // Initialize virtual interface

    endfunction

    task run();

        $display("[%0tps] Driver: Starting...", $time);

        forever begin

            Packet pkt = new(); // Create new packet instance

            mbx.get(pkt); // Retrieve packet from mailbox

            drive(pkt); // Drive packet data to DUT

            vif.rd\_en\_0 = pkt.rd\_en\_0; // Set read enable signals

            vif.rd\_en\_1 = pkt.rd\_en\_1;

            vif.rd\_en\_2 = pkt.rd\_en\_2;

            ->drv\_done; // Signal driver completion

        end

    endtask

    task drive(Packet pkt);

        case (pkt.pkt\_type)

            RESET: reset\_dut(); // Handle reset packet

            HEADER: drive\_header(pkt); // Handle header packet

            PAYLOAD: drive\_payload(pkt); // Handle payload packet

            PARITY: drive\_parity(pkt); // Handle parity packet

            default: $display("Invalid packet type"); // Handle invalid packet types

        endcase

    endtask

    task reset\_dut();

        vif.rst = 0; // Deassert reset

        @(posedge vif.clk);

        vif.rst = 1; // Assert reset

        vif.pkt\_valid = 0; // Deassert packet valid

        @(posedge vif.clk);

    endtask

    task drive\_header(Packet pkt);

        wait(vif.busy == 0); // Wait until DUT is not busy

        @(negedge vif.clk);

        vif.pkt\_valid = 1; // Assert packet valid

        vif.data = pkt.header; // Drive header data

        @(posedge vif.clk);

        @(posedge vif.clk);

    endtask

    task drive\_payload(Packet pkt);

        wait(vif.busy == 0); // Wait until DUT is not busy

        @(negedge vif.clk);

        vif.pkt\_valid <= 1; // Assert packet valid

        vif.data <= pkt.data; // Drive payload data

    endtask

    task drive\_parity(Packet pkt);

        wait(vif.busy == 0); // Wait until DUT is not busy

        @(negedge vif.clk);

        vif.pkt\_valid <= 0; // Deassert packet valid

        vif.data <= pkt.parity; // Drive parity data

    endtask

endclass

## 

## INTERFACE

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: interface.sv                           \*/

/\*      DESCRIPTION: Actual interface definition         \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

interface router\_if();

    logic clk;           // Clock signal

    logic rst;           // Reset signal

    logic [7:0] data;    // Data input

    logic pkt\_valid;     // Packet valid signal

    logic rd\_en\_0;       // Read enable signal for output 0

    logic rd\_en\_1;       // Read enable signal for output 1

    logic rd\_en\_2;       // Read enable signal for output 2

    logic vld\_out\_0;     // Valid output signal for output 0

    logic vld\_out\_1;     // Valid output signal for output 1

    logic vld\_out\_2;     // Valid output signal for output 2

    logic err;           // Error signal

    logic busy;          // Busy signal indicating router activity

    logic [7:0] dout\_0;  // Data output for output 0

    logic [7:0] dout\_1;  // Data output for output 1

    logic [7:0] dout\_2;  // Data output for output 2

    initial clk = 0; // Initialize clock to 0

    always #5 clk = ~clk; // Clock toggles every 5 time units

endinterface

## MONITOR

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: monitor.sv                             \*/

/\*      DESCRIPTION: Mointors the output from the dut     \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

class Monitor;

    local bit[7:0] header = 0; // Stores the header value

    mailbox #(Packet) mbx\_in; // Mailbox for storing incoming packets

    mailbox #(Packet) mbx\_out; // Mailbox for storing outgoing packets

    event drv\_done; // Event to synchronize with the driver

    virtual router\_if vif; // Virtual interface for communication with DUT

    int count = 0; // Counter for incoming packets

    int prev\_val = 0; // Stores previous output value to detect changes

    function new(mailbox #(Packet) mbx\_in, mailbox #(Packet) mbx\_out, event drv\_done, virtual router\_if vif);

        this.mbx\_in = mbx\_in; // Initialize input mailbox

        this.mbx\_out = mbx\_out; // Initialize output mailbox

        this.drv\_done = drv\_done; // Initialize event

        this.vif = vif; // Initialize virtual interface

    endfunction

    task run();

        $display("[%0tps] Monitor: Starting...", $time);

        forever begin

            checkPacket\_in(header); // Check incoming packets

        end

    endtask

    task run1();

        checkPacket\_out(); // Check outgoing packets

    endtask

    task checkPacket\_in(ref bit[7:0] header);

        Packet item = new(); // Create a new packet instance

        @(drv\_done); // Wait for driver completion

        @(posedge vif.clk);

        #1;

        item = parsePacket(); // Parse packet data from interface

        if(!header && item.pkt\_valid) begin

            item.pkt\_type = HEADER; // Identify header packet

            header = item.data;

        end

        else if(header && item.pkt\_valid) begin

            item.pkt\_type = PAYLOAD; // Identify payload packet

        end

        else if(header && !item.pkt\_valid) begin

            item.pkt\_type = PARITY; // Identify parity packet

        end

        else begin

            item.pkt\_type = RESET; // Identify reset packet

        end

        mbx\_in.put(item); // Store packet in input mailbox

        count = count + 1; // Increment packet count

    endtask

    task checkPacket\_out();

        int count\_1 = 0;

        Packet item = new(); // Create a new packet instance

        while(count\_1 < header[7:2] + 1) begin // Process expected number of packets

            @(posedge vif.clk);

            #1;

            wait(vif.rd\_en\_0 || vif.rd\_en\_1 || vif.rd\_en\_2); // Wait for read enable signals

            item = parsePacket(); // Parse packet data from interface

            if(item.rd\_en\_0 && (item.dout\_0 != 0) && (prev\_val != item.dout\_0)) begin

                mbx\_out.put(item); // Store packet in output mailbox

                count\_1 = count\_1 + 1; // Increment processed packet count

                prev\_val = item.dout\_0; // Update previous value

            end

            else if(item.rd\_en\_1 && (item.dout\_1 != 0) && (prev\_val != item.dout\_1)) begin

                mbx\_out.put(item);

                count\_1 = count\_1 + 1;

                prev\_val = item.dout\_1;

            end

            else if (item.rd\_en\_2 && (item.dout\_2 != 0) && (prev\_val != item.dout\_2)) begin

                mbx\_out.put(item);

                count\_1 = count\_1 + 1;

                prev\_val = item.dout\_2;

            end else begin

                count\_1 = count\_1; // Maintain count if no new data

            end

        end

    endtask

    function Packet parsePacket();

        Packet pkt = new(); // Create a new packet instance

        pkt.rst = vif.rst;

        pkt.pkt\_valid = vif.pkt\_valid;

        pkt.data = vif.data;

        pkt.rd\_en\_0 = vif.rd\_en\_0;

        pkt.rd\_en\_1 = vif.rd\_en\_1;

        pkt.rd\_en\_2 = vif.rd\_en\_2;

        pkt.vld\_out\_0 = vif.vld\_out\_0;

        pkt.vld\_out\_1 = vif.vld\_out\_1;

        pkt.vld\_out\_2 = vif.vld\_out\_2;

        pkt.err = vif.err;

        pkt.busy = vif.busy;

        pkt.dout\_0 = vif.dout\_0;

        pkt.dout\_1 = vif.dout\_1;

        pkt.dout\_2 = vif.dout\_2;

        return pkt; // Return parsed packet

    endfunction

endclass

## SCOREBOARD

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*   AUTHOR: METECH                                               \*/

/\*   FILE\_NAME: scoreboard.sv                                       \*/

/\*   DESCRIPTION: Verifies design using the received output and golden reference \*/

/\*   DATE: 03/02/2025                                               \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

class Scoreboard;

    bit[7:0] header = 0; // Stores the packet header

    mailbox #(Packet) mbx\_in; // Mailbox for input packets

    mailbox #(Packet) mbx\_out; // Mailbox for output packets

    virtual router\_if vif; // Virtual interface for router

    bit[7:0] in\_stream[$], out\_stream[$]; // Queues to store input and output data streams

    logic TX\_done = 0; // Transmission done flag

    logic RX\_done = 0; // Reception done flag

    int prev\_val = 0; // Stores previous value to prevent duplicates

    // Constructor: Initializes mailboxes

    function new(mailbox #(Packet) mbx\_in, mailbox #(Packet) mbx\_out);

        this.mbx\_in = mbx\_in;

        this.mbx\_out = mbx\_out;

    endfunction

    // Task to process incoming packets

    task in\_run();

        $display("[%0tps] Scoreboard: Starting...", $time);

        forever begin

            Packet pkt;

            if(mbx\_in.num() > 0) begin

                mbx\_in.get(pkt);

                checkPacket(pkt, this.header);

            end else begin

                #10; // Wait to avoid busy-waiting

            end

        end

    endtask

    // Task to process outgoing packets

    task out\_run();

        int count = 1;

        forever begin

            Packet pkt;

            if (mbx\_out.num() > 0) begin

                mbx\_out.get(pkt);

                checkPacket\_1(pkt);

                checkall();

                if (!(count > header[7:2] + 1)) begin

                   count = count + 1;

                end

            end else begin

                #10; // Prevent busy-waiting

            end

        end

    endtask

    // Task to check incoming packets and store them

    task checkPacket(Packet item, ref bit[7:0] header);

        bit[8:0] cnt;

        if(item.pkt\_valid && item.rst) begin

            if(item.pkt\_type == HEADER) begin

                header = item.data;

                cnt = header[7:2] + 2;

                in\_stream.push\_back(header);

            end

            if(item.pkt\_type == PAYLOAD || item.pkt\_type == PARITY) begin

                in\_stream.push\_back(item.data);

            end

        end

    endtask

    // Task to check outgoing packets and store them

    task checkPacket\_1(Packet item);

        if(item.rd\_en\_0 && (item.dout\_0 != 0) && (prev\_val != item.dout\_0)) begin

            out\_stream.push\_back(item.dout\_0);

            prev\_val = item.dout\_0;

        end

        else if(item.rd\_en\_1 && (item.dout\_1 != 0) && (prev\_val != item.dout\_1)) begin

            out\_stream.push\_back(item.dout\_1);

            prev\_val = item.dout\_1;

        end

        else if(item.rd\_en\_2 && (item.dout\_2 != 0) && (prev\_val != item.dout\_2)) begin

            out\_stream.push\_back(item.dout\_2);

            prev\_val = item.dout\_2;

        end

    endtask

    // Task to compare input and output streams for verification

    task checkall();

        int in\_parity = 0;

        int out\_parity = 0;

        if((in\_stream.size() == header[7:2] + 1) && (out\_stream.size() == header[7:2] + 1)) begin

            foreach(in\_stream[i]) begin

                in\_parity = in\_parity ^ in\_stream[i];

            end

            in\_stream.push\_back(in\_parity);

            foreach(out\_stream[i]) begin

                out\_parity = out\_parity ^ out\_stream[i];

            end

            out\_stream.push\_back(out\_parity);

            if(in\_parity == out\_parity) begin

                $display("/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/");

                $display("/\* Successfully verified Router 1x3");

                $display("/\* Input: %0p", in\_stream);

                $display("/\* Output: %0p", out\_stream);

                $display("/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/");

            end

            else begin

                $display("Verification unsuccessful: in\_parity = %0h, out\_parity = %0h", in\_parity, out\_parity);

            end

        end

    endtask

endclass

## ENVIRONMENT

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                               \*/

/\*      FILE\_NAME: environment.sv                                     \*/

/\*      DESCRIPTION: Connects driver, generator, monitor &scoreboard \*/

/\*      DATE: 03/02/2025                                             \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

`include "transaction.sv"

`include "generator.sv"

`include "driver.sv"

`include "monitor.sv"

`include "scoreboard.sv"

// Environment class to instantiate and connect all verification components

class Environment;

    Generator gen; // Generates test packets

    Driver drv; // Drives packets to DUT

    Monitor mon; // Monitors DUT output

    Scoreboard sbd; // Compares expected vs actual results

    mailbox #(Packet) drv\_mbx; // Mailbox for driver communication

    mailbox #(Packet) sbd\_mbx\_in; // Mailbox for input packets to scoreboard

    mailbox #(Packet) sbd\_mbx\_out; // Mailbox for output packets from scoreboard

    event drv\_done; // Event to synchronize driver completion

    virtual router\_if vif; // Virtual interface for DUT interaction

    function new(virtual router\_if vif);

        drv\_mbx = new(); // Initialize driver mailbox

        sbd\_mbx\_in = new(); // Initialize input mailbox for scoreboard

        sbd\_mbx\_out = new(); // Initialize output mailbox for scoreboard

        gen = new(drv\_mbx, drv\_done); // Instantiate generator

        drv = new(drv\_mbx, drv\_done, vif); // Instantiate driver

        mon = new(sbd\_mbx\_in, sbd\_mbx\_out, drv\_done, vif); // Instantiate monitor

        sbd = new(sbd\_mbx\_in, sbd\_mbx\_out); // Instantiate scoreboard

    endfunction

    task run();

        fork

            gen.run(); // Run generator

            drv.run(); // Run driver

            mon.run(); // Run monitor

            mon.run1(); // Additional monitor function

            sbd.in\_run(); // Run input side of scoreboard

            sbd.out\_run(); // Run output side of scoreboard

        join

    endtask

endclass

## TESTBENCH TOP

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: testbench.sv                         \*/

/\*      DESCRIPTION: Testbench top module                 \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

`include "interface.sv"

`include "environment.sv"

module tb();

    router\_if intf(); // Instantiate the router interface

    Environment env = new(intf); // Create environment instance with the interface

    router dut(intf.clk, intf.rst, intf.data, intf.pkt\_valid, intf.rd\_en\_0, intf.rd\_en\_1, intf.rd\_en\_2, intf.vld\_out\_0, intf.vld\_out\_1, intf.vld\_out\_2, intf.err, intf.busy, intf.dout\_0, intf.dout\_1, intf.dout\_2); // Instantiate the router DUT

    initial begin

        env.run(); // Execute testbench environment

    end

    initial begin

        $dumpfile("out.vcd"); // Specify the VCD file for waveform dumping

        $dumpvars(1); // Dump all variables for debugging

        #2000 $finish; // Terminate simulation after 2000 time units

        end

endmodule

# 

# DESIGN (ROUTER 1x3)

## ROUTER TOP

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: router.sv                             \*/

/\*      DESCRIPTION: Top level module of a 1x3 router   \*/

/\*      DATE: 21/12/2024                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module router (

    input logic clk,                    // Clock input

    input logic rst,                    // Reset input

    input logic [7:0] d\_in,             // Data input (8 bits)

    input logic pkt\_valid,              // Packet validity signal

    input logic rd\_en\_0,                // Read enable for FIFO 0

    input logic rd\_en\_1,                // Read enable for FIFO 1

    input logic rd\_en\_2,                // Read enable for FIFO 2

    output logic vld\_out\_0,             // Valid output for FIFO 0

    output logic vld\_out\_1,             // Valid output for FIFO 1

    output logic vld\_out\_2,             // Valid output for FIFO 2

    output logic err,                   // Error signal

    output logic busy,                  // Busy signal indicating processing

    output logic [7:0] dout\_0,          // Data output from FIFO 0

    output logic [7:0] dout\_1,          // Data output from FIFO 1

    output logic [7:0] dout\_2           // Data output from FIFO 2

);

    // Internal wire declarations for FIFO control signals and state management

    logic soft\_rst\_0, full\_0, empty\_0;

    logic soft\_rst\_1, full\_1, empty\_1;

    logic soft\_rst\_2, full\_2, empty\_2;

    logic fifo\_full, detect\_addr, ld\_state, laf\_state;

    logic full\_state, lfd\_state, rst\_int\_reg;

    logic parity\_done, low\_pkt\_valid, wr\_en\_reg;

    logic [2:0] wr\_en;               // Write enable for the FIFOs

    logic [7:0] din;                 // Data input to FIFOs

    // Instantiate FIFOs

    fifo FIFO\_0 (clk, rst, soft\_rst\_0, wr\_en[0], rd\_en\_0, lfd\_state, din, full\_0, empty\_0, dout\_0);

    fifo FIFO\_1 (clk, rst, soft\_rst\_1, wr\_en[1], rd\_en\_1, lfd\_state, din, full\_1, empty\_1, dout\_1);

    fifo FIFO\_2 (clk, rst, soft\_rst\_2, wr\_en[2], rd\_en\_2, lfd\_state, din, full\_2, empty\_2, dout\_2);

    // Instantiate synchronizer to manage input data and FIFO states

    synchronizer SYNC (

        clk, rst, d\_in[1:0], detect\_addr,

        full\_0, full\_1, full\_2,

        empty\_0, empty\_1, empty\_2,

        wr\_en\_reg, rd\_en\_0, rd\_en\_1, rd\_en\_2,

        wr\_en, fifo\_full,

        vld\_out\_0, vld\_out\_1, vld\_out\_2,

        soft\_rst\_0, soft\_rst\_1, soft\_rst\_2

    );

    // Instantiate registers to store data and manage errors

    register REG\_0 (

        clk, rst, pkt\_valid, d\_in,

        fifo\_full, detect\_addr,

        ld\_state, laf\_state, full\_state, lfd\_state,

        rst\_int\_reg, din, err,

        parity\_done, low\_pkt\_valid

    );

    // Instantiate FSM controller to manage router states and operations

    fsm\_controller FSM (

        clk, rst, pkt\_valid, fifo\_full,

        empty\_0, empty\_1, empty\_2,

        soft\_rst\_0, soft\_rst\_1, soft\_rst\_2,

        parity\_done, low\_pkt\_valid,

        d\_in[1:0], wr\_en\_reg,

        detect\_addr, ld\_state, laf\_state,

        lfd\_state, full\_state,

        rst\_int\_reg, busy

    );

endmodule

## FSM\_CONTROLLER

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: fsm\_controller.sv                     \*/

/\*      DESCRIPTION:  FSM Controller module             \*/

/\*      DATE: 21/12/2024                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module fsm\_controller (

    input logic clk,              // Clock input

    input logic rst,              // Active-low reset signal

    input logic pkt\_valid,        // Signal indicating a valid packet is present

    input logic fifo\_full,        // Signal indicating FIFO is full

    input logic fifo\_empty\_0,     // Signal indicating FIFO 0 is empty

    input logic fifo\_empty\_1,     // Signal indicating FIFO 1 is empty

    input logic fifo\_empty\_2,     // Signal indicating FIFO 2 is empty

    input logic soft\_rst\_0,       // Soft reset signal for FIFO 0

    input logic soft\_rst\_1,       // Soft reset signal for FIFO 1

    input logic soft\_rst\_2,       // Soft reset signal for FIFO 2

    input logic parity\_done,      // Signal indicating parity check is complete

    input logic low\_pkt\_valid,    // Signal indicating a low-valid packet condition

    input logic[1:0] din,        // 2-bit input specifying the destination FIFO

    output logic wr\_en\_req,       // Write enable request signal

    output logic detect\_addr,     // Signal to detect packet address

    output logic ld\_state,        // Load data state indicator

    output logic laf\_state,       // Load after full state indicator

    output logic lfd\_state,       // Load first data state indicator

    output logic full\_state,      // FIFO full state indicator

    output logic rst\_int\_reg,     // Reset internal register signal

    output logic busy             // Busy signal indicating FSM activity

);

  // State encoding for the FSM (1x3 router control)

  parameter DECODE\_ADDRESS     = 3'b000; // State to decode packet address

  parameter LOAD\_FIRST\_DATA    = 3'b001; // State to load the first data word

  parameter LOAD\_DATA          = 3'b010; // State to load subsequent data words

  parameter WAIT\_TILL\_EMPTY    = 3'b011; // Wait for the target FIFO to become empty

  parameter CHECK\_PARITY\_ERROR = 3'b100; // State to check for parity errors

  parameter LOAD\_PARITY        = 3'b101; // Load parity word

  parameter FIFO\_FULL\_STATE    = 3'b110; // State when FIFO is full

  parameter LOAD\_AFTER\_FULL    = 3'b111; // Load data after the FIFO becomes non-full

  logic [2:0] PS, NS; // Current State (PS) and Next State (NS) registers

  // State transition logic triggered on the rising edge of the clock

  always @(posedge clk) begin

    if (!rst)

      PS <= DECODE\_ADDRESS; // Reset state to DECODE\_ADDRESS

    else if (soft\_rst\_0 || soft\_rst\_1 || soft\_rst\_2)

      PS <= DECODE\_ADDRESS; // On any soft reset, transition to DECODE\_ADDRESS

    else

      PS <= NS; // Transition to the next state

  end

  // Next state logic based on the current state and input conditions

  always @(\*) begin

    NS = DECODE\_ADDRESS; // Default next state

    case (PS)

      DECODE\_ADDRESS: begin

        if ((pkt\_valid && din == 0 && fifo\_empty\_0) ||

            (pkt\_valid && din == 1 && fifo\_empty\_1) ||

            (pkt\_valid && din == 2 && fifo\_empty\_2))

          NS = LOAD\_FIRST\_DATA; // Load first data if FIFO is empty

        else if ((pkt\_valid && din == 0 && ~fifo\_empty\_0) ||

                 (pkt\_valid && din == 1 && !fifo\_empty\_1) ||

                 (pkt\_valid && din == 2 && !fifo\_empty\_2))

          NS = WAIT\_TILL\_EMPTY; // Wait if target FIFO is not empty

        else

          NS = DECODE\_ADDRESS; // Stay in the current state

      end

      LOAD\_FIRST\_DATA: NS = LOAD\_DATA; // Transition to LOAD\_DATA state

      LOAD\_DATA: begin

        if (fifo\_full)

          NS = FIFO\_FULL\_STATE; // If FIFO is full, transition to full state

        else if (!fifo\_full && !pkt\_valid)

          NS = LOAD\_PARITY; // If no more data, load parity

        else

          NS = LOAD\_DATA; // Continue loading data

      end

      WAIT\_TILL\_EMPTY: begin

        if (fifo\_empty\_0 || fifo\_empty\_1 || fifo\_empty\_2)

          NS = LOAD\_FIRST\_DATA; // If any FIFO becomes empty, load first data

        else

          NS = WAIT\_TILL\_EMPTY; // Continue waiting

      end

      FIFO\_FULL\_STATE: begin

        if (!fifo\_full)

          NS = LOAD\_AFTER\_FULL; // If FIFO is no longer full, load after full

        else

          NS = FIFO\_FULL\_STATE; // Stay in the full state

      end

      LOAD\_AFTER\_FULL: begin

        if (!parity\_done && !low\_pkt\_valid)

          NS = LOAD\_DATA; // If parity not done and valid, load data

        else if (!parity\_done && low\_pkt\_valid)

          NS = LOAD\_PARITY; // If low packet valid, load parity

        else if (parity\_done)

          NS = DECODE\_ADDRESS; // If parity done, decode next address

      end

      LOAD\_PARITY: NS = CHECK\_PARITY\_ERROR; // Transition to parity check

      CHECK\_PARITY\_ERROR: begin

        if (fifo\_full)

          NS = FIFO\_FULL\_STATE; // If FIFO is full, go to full state

        else

          NS = DECODE\_ADDRESS; // Otherwise, decode next address

      end

    endcase

  end

  // Output assignments based on the current state

  assign detect\_addr = (PS == DECODE\_ADDRESS); // Detect address in decode state

  assign wr\_en\_req = (PS == LOAD\_DATA || PS == LOAD\_PARITY || PS == LOAD\_AFTER\_FULL); // Write enable in specific states

  assign full\_state = (PS == FIFO\_FULL\_STATE); // Indicate FIFO full state

  assign lfd\_state = (PS == LOAD\_FIRST\_DATA); // Indicate load first data state

  assign busy = (PS == LOAD\_FIRST\_DATA || PS == LOAD\_PARITY || PS == FIFO\_FULL\_STATE || PS == LOAD\_AFTER\_FULL || PS == WAIT\_TILL\_EMPTY || PS == CHECK\_PARITY\_ERROR); // Indicate FSM is busy

  assign ld\_state = (PS == LOAD\_DATA); // Indicate load data state

  assign laf\_state = (PS == LOAD\_AFTER\_FULL); // Indicate load after full state

  assign rst\_int\_reg = (PS == CHECK\_PARITY\_ERROR); // Reset internal register during parity check

endmodule

## REGISTER

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                  \*/

/\*      FILE\_NAME: register.sv                           \*/

/\*      DESCRIPTION:  register module                   \*/

/\*      DATE: 21/12/2024                                \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module register (

    input logic clk,                // Clock input

    input logic rst,                // Active-low reset signal

    input logic pkt\_valid,          // Packet valid signal indicating data validity

    input logic [7:0] din,          // 8-bit data input

    input logic fifo\_full,          // Signal indicating if the FIFO is full

    input logic detect\_addr,        // Signal for address detection

    input logic ld\_state,           // Signal indicating load state is active

    input logic laf\_state,          // Signal indicating load after full state

    input logic full\_state,         // Signal indicating the full state of the system

    input logic lfd\_state,          // Signal indicating load first data state

    input logic rst\_int\_reg,        // Signal to reset the internal register

    output logic [7:0] dout,    // 8-bit data output

    output logic err,           // Error signal output

    output logic parity\_done,   // Parity check completion flag

    output logic low\_pkt\_valid  // Signal indicating low packet validity

);

  // Internal registers to store data, parity, and intermediate values

  logic [7:0] header, int\_reg, int\_parity, ext\_parity;

  // PARITY DONE LOGIC: Controls when the parity check is marked as done

  always @(posedge clk) begin

    if (!rst)

      parity\_done <= 0;       // Reset parity done flag

    else if (detect\_addr)

      parity\_done <= 0;       // Reset if address detection occurs

    else if ((ld\_state && (~fifo\_full) && (~pkt\_valid)) ||

             (laf\_state && low\_pkt\_valid && (~parity\_done)))

      parity\_done <= 1;       // Set parity done if conditions are met

  end

  // LOW PACKET VALID LOGIC: Manages the `low\_pkt\_valid` flag

  always @(posedge clk) begin

    if (!rst)

      low\_pkt\_valid <= 0;     // Reset low packet valid flag

    else if (rst\_int\_reg)

      low\_pkt\_valid <= 0;     // Reset if internal register is reset

    else if (ld\_state && ~pkt\_valid)

      low\_pkt\_valid <= 1;     // Set if in load state and no valid packet

  end

  // DATA OUT LOGIC: Controls the data output based on various states

  always @(posedge clk) begin

    if (!rst) begin

      dout <= 0;              // Reset data output

      header <= 0;            // Reset header register

      int\_reg <= 0;           // Reset internal register

    end else if (detect\_addr && pkt\_valid && din[1:0] != 2'b11)

      header <= din;          // Capture header if address is detected and packet is valid

    else if (lfd\_state)

      dout <= header;         // Output header if in load first data state

    else if (ld\_state && ~fifo\_full)

      dout <= din;            // Output data if in load state and FIFO is not full

    else if (ld\_state && fifo\_full)

      int\_reg <= din;         // Store data in internal register if FIFO is full

    else if (laf\_state)

      dout <= int\_reg;        // Output internal register data if in load after full state

  end

  // PARITY CALCULATE LOGIC: Computes the internal parity for error checking

  always @(posedge clk) begin

    if (!rst)

      int\_parity <= 0;      // Reset internal parity

    else if (detect\_addr)

      int\_parity <= 0;      // Reset if address detection occurs

    else if (lfd\_state && pkt\_valid)

      int\_parity <= int\_parity ^ header; // XOR with header data if packet is valid

    else if (ld\_state && pkt\_valid && ~full\_state)

      int\_parity <= int\_parity ^ din; // XOR with data input if in load state

    else

      int\_parity <= int\_parity; // Hold current parity value

  end

  // ERROR LOGIC: Checks if there is a parity error

  always @(posedge clk) begin

    if (!rst)

      err <= 0; // Reset error flag

    else if (parity\_done) begin

      if (int\_parity == ext\_parity)

        err <= 0; // No error if internal and external parity match

      else

        err <= 1; // Set error if parities do not match

    end else

      err <= 0; // Hold error as 0 if parity is not done

  end

  // EXTERNAL PARITY LOGIC: Stores the external parity value

  always @(posedge clk) begin

    if (!rst)

      ext\_parity <= 0; // Reset external parity

    else if (detect\_addr)

      ext\_parity <= 0; // Reset if address detection occurs

    else if ((ld\_state && !fifo\_full && !pkt\_valid) ||

             (laf\_state && ~parity\_done && low\_pkt\_valid))

      ext\_parity <= din; // Store data input as external parity if conditions are met

  end

endmodule

## FIFO

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: fifo.sv                               \*/

/\*      DESCRIPTION:  16x9 Fifo Module                   \*/

/\*      DATE: 21/12/2024                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module fifo (

    input logic clk,             // Clock input signal

    input logic rst,             // Active-low reset signal

    input logic soft\_reset,      // Soft reset signal to clear certain outputs

    input logic wr\_en,           // Write enable signal

    input logic rd\_en,           // Read enable signal

    input logic lfd\_state,       // State indicating the first data word (header)

    input logic [7:0] din,       // 8-bit input data to write into the FIFO

    output logic full,           // Full flag indicating the FIFO is full

    output logic empty,          // Empty flag indicating the FIFO is empty

    output logic [7:0] dout  // 8-bit output data from the FIFO

);

  logic [4:0] rd\_ptr, wr\_ptr;    // Read and write pointers (5 bits to track overflow)

  logic [6:0] intCount;          // Counter for tracking multi-byte packet size

  logic [8:0] mem[15:0];         // Memory array with 9-bit width for data + header bit                   //

  logic lfd\_state\_t;             // Temporary register to hold the lfd\_state signal

  // Latching lfd\_state signal with each clock cycle

  always @(posedge clk) begin

    if (!rst)

      lfd\_state\_t <= 0;        // Reset lfd\_state\_t when reset is active

    else

      lfd\_state\_t <= lfd\_state; // Store the current lfd\_state value

  end

  // Managing data output based on read enable and empty status

  always @(posedge clk) begin

    if (!rst)

      dout <= 8'b0;            // Reset dout to 0 on reset

    else if (soft\_reset)

      dout <= 8'bz;            // Set dout to high impedance on soft reset

    else if (rd\_en && !empty)

      dout <= mem[rd\_ptr[3:0]][7:0]; // Read data from memory if enabled and not empty

    else if (intCount == 0)

      dout <= 8'bz;            // High impedance when no data to output

  end

  // Memory write logic: Writing input data into the FIFO

  always @(posedge clk) begin

    if (!rst || soft\_reset) begin

      // Reset all memory locations on reset or soft reset

      for (int i = 0; i < 16; i = i + 1)

        mem[i] <= 0;

    end else if (wr\_en && !full) begin

      // Write data into memory if write enabled and not full

      if (lfd\_state\_t) begin

        mem[wr\_ptr[3:0]][8] <= 1'b1;  // Mark as header word

        mem[wr\_ptr[3:0]][7:0] <= din; // Store input data

      end else begin

        mem[wr\_ptr[3:0]][8] <= 1'b0;  // Mark as regular data word

        mem[wr\_ptr[3:0]][7:0] <= din; // Store input data

      end

    end

  end

  // Write pointer update logic

  always @(posedge clk) begin

    if (!rst)

      wr\_ptr <= 0;             // Reset write pointer

    else if (wr\_en && !full)

      wr\_ptr <= wr\_ptr + 1;    // Increment write pointer on write enable

  end

  // Read pointer update logic

  always @(posedge clk) begin

    if (!rst)

      rd\_ptr <= 0;             // Reset read pointer

    else if (rd\_en && !empty)

      rd\_ptr <= rd\_ptr + 1;    // Increment read pointer on read enable

  end

  // Internal counter management for tracking data packets

  always @(posedge clk) begin

    if (rd\_en && !empty) begin

      // If header word, initialize intCount with data size + 1

      if (mem[rd\_ptr[3:0]][8] == 1'b1)

        intCount <= mem[rd\_ptr[3:0]][7:2] + 1'b1;

      // Otherwise, decrement intCount if it's not zero

      else if (intCount != 0)

        intCount <= intCount - 1'b1;

    end

  end

  // Full flag: Set when write and read pointers overlap with different MSBs

  assign full = (wr\_ptr == {~rd\_ptr[4], rd\_ptr[3:0]});

  // Empty flag: Set when write and read pointers are identical

  assign empty = (rd\_ptr == wr\_ptr);

endmodule

## SYNCHRONIZER

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: synchronizer.sv                       \*/

/\*      DESCRIPTION:  Synchronizer Module               \*/

/\*      DATE: 21/12/2024                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module synchronizer (

    input logic clk,              // Clock input signal

    input logic rst,              // Active-low reset signal

    input logic[1:0] din,        // 2-bit input data to determine FIFO selection

    input logic detect\_addr,      // Signal indicating if address detection is active

    input logic full\_0,           // Signal indicating if FIFO 0 is full

    input logic full\_1,           // Signal indicating if FIFO 1 is full

    input logic full\_2,           // Signal indicating if FIFO 2 is full

    input logic empty\_0,          // Signal indicating if FIFO 0 is empty

    input logic empty\_1,          // Signal indicating if FIFO 1 is empty

    input logic empty\_2,          // Signal indicating if FIFO 2 is empty

    input logic wr\_en\_reg,        // Write enable register input

    input logic rd\_en\_0,          // Read enable signal for FIFO 0

    input logic rd\_en\_1,          // Read enable signal for FIFO 1

    input logic rd\_en\_2,          // Read enable signal for FIFO 2

    output logic [2:0] wr\_en, // 3-bit output to control write enable for each FIFO

    output logic fifo\_full,   // Output indicating if the selected FIFO is full

    output logic vld\_out\_0,       // Valid output signal for FIFO 0 (not empty)

    output logic vld\_out\_1,       // Valid output signal for FIFO 1 (not empty)

    output logic vld\_out\_2,       // Valid output signal for FIFO 2 (not empty)

    output logic soft\_reset\_0,// Soft reset signal for FIFO 0

    output logic soft\_reset\_1,// Soft reset signal for FIFO 1

    output logic soft\_reset\_2 // Soft reset signal for FIFO 2

);

  // Registers to count cycles for each FIFO's inactivity

  logic [5:0] count0, count1, count2;

  logic [1:0] tmp\_din; // Temporary register to hold the address input

  // Capture 'din' value on the detection of address

  always @(posedge clk) begin

    if (!rst)

      tmp\_din <= 0; // Reset 'tmp\_din' to 0 when reset is active

    else if (detect\_addr)

      tmp\_din <= din; // Store 'din' if address detection is active

  end

  // Control logic for write enable and FIFO full status based on 'tmp\_din'

  always @(\*) begin

    case (tmp\_din)

      2'b00: begin // Case for FIFO 0

        fifo\_full <= full\_0; // Set full status for FIFO 0

        wr\_en <= (wr\_en\_reg) ? 3'b001 : 0; // Enable write if 'wr\_en\_reg' is set

      end

      2'b01: begin // Case for FIFO 1

        fifo\_full <= full\_1; // Set full status for FIFO 1

        wr\_en <= (wr\_en\_reg) ? 3'b010 : 0; // Enable write if 'wr\_en\_reg' is set

      end

      2'b10: begin // Case for FIFO 2

        fifo\_full <= full\_2; // Set full status for FIFO 2

        wr\_en <= (wr\_en\_reg) ? 3'b100 : 0; // Enable write if 'wr\_en\_reg' is set

      end

      default: begin // Default case: no FIFO selected

        fifo\_full <= 0;

        wr\_en <= 0;

      end

    endcase

  end

  // Assign valid outputs based on FIFO emptiness

  assign vld\_out\_0 = (~empty\_0); // Valid if FIFO 0 is not empty

  assign vld\_out\_1 = (~empty\_1); // Valid if FIFO 1 is not empty

  assign vld\_out\_2 = (~empty\_2); // Valid if FIFO 2 is not empty

  // Monitor FIFO 0 for inactivity and trigger soft reset if needed

  always @(posedge clk) begin

    if (!rst) begin

      count0 <= 0;

      soft\_reset\_0 <= 0; // Reset state for FIFO 0

    end else if (vld\_out\_0) begin // If FIFO 0 has valid data

      if (!rd\_en\_0) begin // If not being read

        if (count0 == 29) begin // After 30 cycles

          soft\_reset\_0 <= 1; // Trigger soft reset

          count0 <= 0; // Reset counter

        end else begin

          soft\_reset\_0 <= 0;

          count0 <= count0 + 1; // Increment counter

        end

      end else

        count0 <= 0; // Reset counter if being read

    end

  end

  // Monitor FIFO 1 for inactivity and trigger soft reset if needed

  always @(posedge clk) begin

    if (!rst) begin

      count1 <= 0;

      soft\_reset\_1 <= 0; // Reset state for FIFO 1

    end else if (vld\_out\_1) begin // If FIFO 1 has valid data

      if (!rd\_en\_1) begin // If not being read

        if (count1 == 29) begin // After 30 cycles

          soft\_reset\_1 <= 1; // Trigger soft reset

          count1 <= 0; // Reset counter

        end else begin

          soft\_reset\_1 <= 0;

          count1 <= count1 + 1; // Increment counter

        end

      end else

        count1 <= 0; // Reset counter if being read

    end

  end

  // Monitor FIFO 2 for inactivity and trigger soft reset if needed

  always @(posedge clk) begin

    if (!rst) begin

      count2 <= 0;

      soft\_reset\_2 <= 0; // Reset state for FIFO 2

    end else if (vld\_out\_2) begin // If FIFO 2 has valid data

      if (!rd\_en\_2) begin // If not being read

        if (count2 == 29) begin // After 30 cycles

          soft\_reset\_2 <= 1; // Trigger soft reset

          count2 <= 0; // Reset counter

        end else begin

          soft\_reset\_2 <= 0;

          count2 <= count2 + 1; // Increment counter

        end

      end else

        count2 <= 0; // Reset counter if being read

    end

  end

endmodule

# OUTPUT

![](data:image/png;base64,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)